Overall Objective

Load Libraries

library(tidyverse)
library(cowplot)
library(broom)
library(plotly)

Import data

Convert data from ‘wide’ to ‘long’ format

# data
data1 <- data %>%
  gather(Sample,Count,2:49)
# Separate samples by identifiers 
data2 <- data1 %>% 
  separate(Sample, into=c("Sample_ID","Dilution_factor",
                          "Injection","Tech_rep", sep = "_")) %>% 
  select(-`_`)
# Standards
standards1 <- standards %>% 
  gather(Sample,Count,2:25)
standards2 <- standards1 %>% 
  separate(Sample, into=c("Sample_ID","When","Dilution_factor",
                          "Nano_day","Injection","Tech_Rep", sep = "_")) %>% 
  select(-`_`)

Factor the data into categorical variables

# Refactoring Columns for samples
data2$Sample_ID <- as.factor(data2$Sample_ID)
data2$Dilution_factor <- as.numeric(data2$Dilution_factor)
data2$Injection<- as.factor(data2$Injection)
data2$Tech_rep <- as.numeric(data2$Tech_rep)
data2
# Refactoring COlumns for key
key$Sample_ID <- as.factor(key$Sample_ID)
key$Time <- as.factor(key$Time)
key$Treatment <- as.factor(key$Treatment)
key$Volume <- as.numeric(key$Volume)
key$Treatment <- factor(key$Treatment,levels = c('DMSO','EGF','BPS','BPS_EGF'))
key
# Refactoring columns for standards
standards2$Sample_ID <- as.factor(standards2$Sample_ID)
standards2$When <- as.factor(standards2$When)
standards2$Dilution_factor <- as.numeric(standards2$Dilution_factor)
standards2$Injection <- as.factor(standards2$Injection)
standards2$Nano_day <- as.numeric(standards2$Nano_day)
standards2

Back calculate standards

standards2 <- standards2 %>% 
  mutate(True_Count=Dilution_factor*Count)
# Set the correct order of 'categorical factors'
standards2$Nano_day <-  factor(standards2$Nano_day, levels=c('1','2'))
standards2$When <- factor(standards2$When, levels=c('before','after'))
standards2

Summarize three technical standard replicates

standards3 <- standards2 %>% 
  group_by(particle_size,Sample_ID,When,Dilution_factor,Nano_day,Injection) %>% 
  summarise( tech_N = length(True_Count),
             tech_mean = mean(True_Count),
             tech_sd = sd(True_Count),
             tech_se = tech_sd/sqrt(tech_N))
standards3

Summarize standards by injection

standards4 <- standards3 %>% 
  group_by(Nano_day,When,particle_size) %>% 
  summarise( inj_N = length(tech_mean),
             inj_mean = mean(tech_mean),
             inj_sd = sd(tech_mean),
             inj_se = inj_sd/sqrt(inj_N))
standards4

Plot before and after plots, facet by experimental day

std_plot <- standards4 %>% 
  ggplot(aes(x = particle_size, y = inj_mean, color=When))+
  geom_line(size=2) + xlim(0,300)+ #line size, x-axis scale
  geom_ribbon(aes(ymin=inj_mean-inj_se, ymax=inj_mean+inj_se),
              alpha=0.4,fill = alpha('grey12', 0.2)) + #error bars
  scale_y_continuous(expand=c(0,0))+ #set bottom of graph
  xlab("Particle Size") + # X axis label
  ylab("\nMean Particle Concentration/ml\n") + # Y axis label
  ggtitle("Nanosight Histogram of\n100nm Standards")+ #title
  labs(color="Condition")+ #Label table title
  facet_wrap(~ Nano_day)
std_plot

# ggsave("Standards_histogram_plot.png",
#        height = 5, width = 7, dpi = 300, units= "in")

Standards particle concentrations from each experimental day

standards_df <- standards4 %>% 
  group_by(Nano_day,When) %>% 
  summarise(total=sum(inj_mean))
standards_df

Bar graph of standards particle concentrations

standards_bar <- standards_df %>% 
  ggplot(aes(x=Nano_day,y=total,fill=When))+
  geom_col(position="dodge")+
  scale_y_continuous(expand=c(0,0))+ #set bottom of graph
  xlab("Experimental Day") + # X axis label
  ylab("\nMean Particle Concentration/ml\n") + # Y axis label
  ggtitle("Nanosight Histogram of\n100nm Standards")+ #title
  labs(color="When") #Label table title
standards_bar

# ggsave("Standards_bar_plot.png",
#        height = 5, width = 7, dpi = 300, units= "in")

Intraassay variability

Intra.assay_cv <- standards_df %>% 
  group_by(Nano_day) %>% 
  summarise(Intra_Day_N = length(total),
            Intra_Day_mean = mean(total),
            Intra_Day_sd = sd(total),
            Intra_Day_se = Intra_Day_sd/sqrt(Intra_Day_N),
            Intra_Day_cv = Intra_Day_sd/Intra_Day_mean )
Intra.assay_cv
# # Save as .csv
# write_csv(Intra.assay_cv,"Intra.assay_cv.csv")

Interassay variability

Inter.assay_cv <- Intra.assay_cv %>% 
  summarise(Inter_Day_N = length(Intra_Day_mean),
            Inter_Day_mean = mean(Intra_Day_mean),
            Inter_Day_sd = sd(Intra_Day_mean),
            Inter_Day_se = Inter_Day_sd/sqrt(Inter_Day_N),
            Inter_Day_cv = Inter_Day_sd/Inter_Day_mean )
Inter.assay_cv
# # Save as .csv
# write_csv(Inter.assay_cv,"Inter.assay_cv.csv")

Sample analysis

Back calculate the original concentration of the sample

data2 <- data2 %>% 
  mutate(True_Count = Dilution_factor*Count)
data2

Average three technical readings

data3 <- data2 %>% 
  group_by(particle_size,Sample_ID,Dilution_factor,Injection) %>% 
  summarise( tech_N = length(True_Count),
             tech_mean = mean(True_Count),
             tech_sd = sd(True_Count),
             tech_se = tech_sd/sqrt(tech_N))
data3

Summarize samples by injection (average both injections)

data4 <- data3 %>% 
  group_by(particle_size,Sample_ID,Dilution_factor) %>% 
  summarise( inj_N = length(tech_mean),
             inj_mean = mean(tech_mean),
             inj_sd = sd(tech_mean),
             inj_se = inj_sd/sqrt(inj_N))
data4
# Average technical replicates and merge with key
merge <- left_join(key,data3, by= "Sample_ID")
merge
# Save as .csv
# write_csv(merge,"Technical_replicate_average.csv")
 
# Average injection replicates and merge with key
merge1 <- left_join(key,data4, by= "Sample_ID")
merge1
# #Save as .csv
# write_csv(merge1,"Injection_replicate_average.csv")

Quick visualizations

Graphing all samples

sample_plot <- merge %>%
  ggplot(aes(x=particle_size, y=tech_mean,color=Injection ))+ #plot
  geom_ribbon(aes(ymin=tech_mean-tech_se,
                  ymax=tech_mean+tech_se),
                  alpha=0.2,fill = alpha('grey12', 0.2)) + #error bars
  geom_line(size=2.0, alpha = 0.8) + xlim(0,500)+ #line size, x-axis scale
  scale_y_continuous(expand=c(0,0))+ #set bottom of graph
  xlab("Particle Size") + # X axis label
  ylab("\nMean Particle Concentration/ml\n") + # Y axis label
  ggtitle("Nanosight Histogram of\nhCTBS treated with BPS")+ #title
  labs(color="Injection")+ #Label table title
  facet_grid(Time ~ Treatment)
  # geom_vline(xintercept = 200)+
  # annotate("text", x= 350, y = 1E8, label= "200nm")
sample_plot

# ggsave("Nanosight_Sample_Histogram.png", plot = sample_plot,
#        height = 10, width = 14, dpi = 200, units= "in")

Interactive Plot

ggplotly(sample_plot)

Particle concentration values for each of the samples

merge2 <- merge1 %>% 
  group_by(Time, Treatment, Volume) %>% 
  summarise(particle_conc=sum(inj_mean))
merge2

Correct for resuspension volume

merge3 <- merge2 %>% 
  mutate(particle_count = (Volume/1000)*particle_conc, # Create new column with number of particles
         corrected_particle_conc = (particle_conc/1E9)) # Create new column with correct particle concentration
merge3
# Save as .csv
# write_csv(merge3,"Adjusted_particle_concentration.csv")

Barplot

plot1 <- merge3 %>% 
  ggplot(aes(x = Treatment, y = corrected_particle_conc, fill = Treatment)) +
  geom_bar(aes(text = paste("Particle Concentration:",
                            corrected_particle_conc)),
           stat="identity", position = "dodge")+
  xlab("\nTreatment\n") + # X axis label
  ylab("\nMean Vessicle Concentration\n(10^9 particles/ ml)\n") + # Y axis label
  ggtitle("Effect of BPS on Extracellular Vessicle\nRelease of hCTBs\n")+
  scale_y_continuous(breaks = seq(0,14,2),
                     limits = c(0,14),
                     expand = c(0,0))+ # set bottom of graph
  labs(color="Condition")+ # Label table title
  facet_wrap(~Time)
Ignoring unknown aesthetics: text
  
plot1

# ggsave("BPS_treated_hCTBs_48_96_facet_plot.png",
#       height = 8, width = 10, dpi = 600, units= "in")

Interactive Plot

ggplotly(plot1)

Barplot

plot2 <- merge3 %>%
  group_by(Time) %>% 
  ggplot(aes(x = Treatment, y = corrected_particle_conc, fill = Time )) +
  geom_bar(position = "dodge", stat = "identity")+
  xlab("\nTreatment\n") + # X axis label
  ylab("\nMean Vessicle Concentration\n(10^9 particles/ ml)\n") + # Y axis label
  ggtitle("Effect of BPS on Extracellular Vessicle\nRelease of hCTBs\n")+
  scale_y_continuous(breaks = seq(0,14,2),
                     limits = c(0,14),
                     expand = c(0,0))+ # set bottom of graph
  labs(fill= "Time (hr)")
plot2

 # ggsave("BPS_treated_hCTBs_48_96_plot.png",
 #       height = 5, width = 7, dpi = 600, units= "in")

Interactive Plot

ggplotly(plot2)

Statistics

fit <- aov(corrected_particle_conc ~ Time * Treatment ,data=merge3)
tidy(fit)
LS0tCnRpdGxlOiAiSHVtYW4gQ1RCIEJQUyBQcm9qZWN0IgphdXRob3I6ICJTZWFuIE5ndXllbiIKb3V0cHV0OgogIGh0bWxfZG9jdW1lbnQ6IAogICAgZGZfcHJpbnQ6IHBhZ2VkCiAgICB0aGVtZTogZmxhdGx5CiAgaHRtbF9ub3RlYm9vazoKICAgIGRmX3ByaW50OiBwYWdlZAogIHBkZl9kb2N1bWVudDogZGVmYXVsdAotLS0KI092ZXJhbGwgT2JqZWN0aXZlIAoKCiMjTG9hZCBMaWJyYXJpZXMKCmBgYHtyLCBlY2hvPVRSVUUsIG1lc3NhZ2U9RkFMU0UsIHdhcm5pbmc9RkFMU0V9CmxpYnJhcnkodGlkeXZlcnNlKQpsaWJyYXJ5KGNvd3Bsb3QpCmxpYnJhcnkoYnJvb20pCmxpYnJhcnkocGxvdGx5KQoKYGBgCgojI0ltcG9ydCBkYXRhCmBgYHtyLCBtZXNzYWdlPUZBTFNFLCB3YXJuaW5nPUZBTFNFLCBpbmNsdWRlPUZBTFNFfQojc2V0d2QoIn4vR2l0SHViL0V4cGVyaW1lbnRzL0NUQl9CUFMvZGF0YS8iKQpzZXR3ZCgifi9MaWJyYXJ5L01vYmlsZSBEb2N1bWVudHMvY29tfmFwcGxlfkNsb3VkRG9jcy9FeHBlcmltZW50cy9DVEJfQlBTL2RhdGEvIikKCgpyYXdfZGF0YSA8LSAiQ1RCX3BpbG90XzQ4aHJfOTZoci1FeHBlcmltZW50U3VtbWFyeS5jc3YiCmtleV9kYXRhIDwtICJLZXlfaENUQlBldHJvZmZfTG9wZXouY3N2IgpzdGFuZGFyZF9kYXRhIDwtICJzdGRfQ1RCX3BpbG90XzQ4aHJfOTZoci1FeHBlcmltZW50U3VtbWFyeS5jc3YiCgoKZGF0YSA8LSByZWFkX2NzdihyYXdfZGF0YSkKc3RhbmRhcmRzIDwtIHJlYWRfY3N2KHN0YW5kYXJkX2RhdGEpCmtleSA8LSByZWFkX2NzdihrZXlfZGF0YSkKCmBgYAoKIyNDb252ZXJ0IGRhdGEgZnJvbSAnd2lkZScgdG8gJ2xvbmcnIGZvcm1hdApgYGB7ciwgbWVzc2FnZT1GQUxTRSwgd2FybmluZz1GQUxTRX0KIyBkYXRhCmRhdGExIDwtIGRhdGEgJT4lCiAgZ2F0aGVyKFNhbXBsZSxDb3VudCwyOjQ5KQoKIyBTZXBhcmF0ZSBzYW1wbGVzIGJ5IGlkZW50aWZpZXJzIApkYXRhMiA8LSBkYXRhMSAlPiUgCiAgc2VwYXJhdGUoU2FtcGxlLCBpbnRvPWMoIlNhbXBsZV9JRCIsIkRpbHV0aW9uX2ZhY3RvciIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgIkluamVjdGlvbiIsIlRlY2hfcmVwIiwgc2VwID0gIl8iKSkgJT4lIAogIHNlbGVjdCgtYF9gKQoKIyBTdGFuZGFyZHMKc3RhbmRhcmRzMSA8LSBzdGFuZGFyZHMgJT4lIAogIGdhdGhlcihTYW1wbGUsQ291bnQsMjoyNSkKCgpzdGFuZGFyZHMyIDwtIHN0YW5kYXJkczEgJT4lIAogIHNlcGFyYXRlKFNhbXBsZSwgaW50bz1jKCJTYW1wbGVfSUQiLCJXaGVuIiwiRGlsdXRpb25fZmFjdG9yIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAiTmFub19kYXkiLCJJbmplY3Rpb24iLCJUZWNoX1JlcCIsIHNlcCA9ICJfIikpICU+JSAKICBzZWxlY3QoLWBfYCkKCgpgYGAKCgojI0ZhY3RvciB0aGUgZGF0YSBpbnRvIGNhdGVnb3JpY2FsIHZhcmlhYmxlcwpgYGB7ciwgd2FybmluZz1GQUxTRX0KIyBSZWZhY3RvcmluZyBDb2x1bW5zIGZvciBzYW1wbGVzCmRhdGEyJFNhbXBsZV9JRCA8LSBhcy5mYWN0b3IoZGF0YTIkU2FtcGxlX0lEKQpkYXRhMiREaWx1dGlvbl9mYWN0b3IgPC0gYXMubnVtZXJpYyhkYXRhMiREaWx1dGlvbl9mYWN0b3IpCmRhdGEyJEluamVjdGlvbjwtIGFzLmZhY3RvcihkYXRhMiRJbmplY3Rpb24pCmRhdGEyJFRlY2hfcmVwIDwtIGFzLm51bWVyaWMoZGF0YTIkVGVjaF9yZXApCgoKIyBSZWZhY3RvcmluZyBDT2x1bW5zIGZvciBrZXkKa2V5JFNhbXBsZV9JRCA8LSBhcy5mYWN0b3Ioa2V5JFNhbXBsZV9JRCkKa2V5JFRpbWUgPC0gYXMuZmFjdG9yKGtleSRUaW1lKQprZXkkVHJlYXRtZW50IDwtIGFzLmZhY3RvcihrZXkkVHJlYXRtZW50KQprZXkkVm9sdW1lIDwtIGFzLm51bWVyaWMoa2V5JFZvbHVtZSkKCmtleSRUcmVhdG1lbnQgPC0gZmFjdG9yKGtleSRUcmVhdG1lbnQsbGV2ZWxzID0gYygnRE1TTycsJ0VHRicsJ0JQUycsJ0JQU19FR0YnKSkKCgojIFJlZmFjdG9yaW5nIGNvbHVtbnMgZm9yIHN0YW5kYXJkcwpzdGFuZGFyZHMyJFNhbXBsZV9JRCA8LSBhcy5mYWN0b3Ioc3RhbmRhcmRzMiRTYW1wbGVfSUQpCnN0YW5kYXJkczIkV2hlbiA8LSBhcy5mYWN0b3Ioc3RhbmRhcmRzMiRXaGVuKQpzdGFuZGFyZHMyJERpbHV0aW9uX2ZhY3RvciA8LSBhcy5udW1lcmljKHN0YW5kYXJkczIkRGlsdXRpb25fZmFjdG9yKQpzdGFuZGFyZHMyJEluamVjdGlvbiA8LSBhcy5mYWN0b3Ioc3RhbmRhcmRzMiRJbmplY3Rpb24pCnN0YW5kYXJkczIkTmFub19kYXkgPC0gYXMubnVtZXJpYyhzdGFuZGFyZHMyJE5hbm9fZGF5KQoKYGBgCgoKCiMjI0JhY2sgY2FsY3VsYXRlIHN0YW5kYXJkcwpgYGB7cn0Kc3RhbmRhcmRzMiA8LSBzdGFuZGFyZHMyICU+JSAKICBtdXRhdGUoVHJ1ZV9Db3VudD1EaWx1dGlvbl9mYWN0b3IqQ291bnQpCgojIFNldCB0aGUgY29ycmVjdCBvcmRlciBvZiAnY2F0ZWdvcmljYWwgZmFjdG9ycycKc3RhbmRhcmRzMiROYW5vX2RheSA8LSAgZmFjdG9yKHN0YW5kYXJkczIkTmFub19kYXksIGxldmVscz1jKCcxJywnMicpKQpzdGFuZGFyZHMyJFdoZW4gPC0gZmFjdG9yKHN0YW5kYXJkczIkV2hlbiwgbGV2ZWxzPWMoJ2JlZm9yZScsJ2FmdGVyJykpCgpgYGAKCiMjI1N1bW1hcml6ZSB0aHJlZSB0ZWNobmljYWwgc3RhbmRhcmQgcmVwbGljYXRlcwpgYGB7cn0Kc3RhbmRhcmRzMyA8LSBzdGFuZGFyZHMyICU+JSAKICBncm91cF9ieShwYXJ0aWNsZV9zaXplLFNhbXBsZV9JRCxXaGVuLERpbHV0aW9uX2ZhY3RvcixOYW5vX2RheSxJbmplY3Rpb24pICU+JSAKICBzdW1tYXJpc2UoIHRlY2hfTiA9IGxlbmd0aChUcnVlX0NvdW50KSwKICAgICAgICAgICAgIHRlY2hfbWVhbiA9IG1lYW4oVHJ1ZV9Db3VudCksCiAgICAgICAgICAgICB0ZWNoX3NkID0gc2QoVHJ1ZV9Db3VudCksCiAgICAgICAgICAgICB0ZWNoX3NlID0gdGVjaF9zZC9zcXJ0KHRlY2hfTikpCnN0YW5kYXJkczMKYGBgCgoKIyMjU3VtbWFyaXplIHN0YW5kYXJkcyBieSBpbmplY3Rpb24KYGBge3J9CnN0YW5kYXJkczQgPC0gc3RhbmRhcmRzMyAlPiUgCiAgZ3JvdXBfYnkoTmFub19kYXksV2hlbixwYXJ0aWNsZV9zaXplKSAlPiUgCiAgc3VtbWFyaXNlKCBpbmpfTiA9IGxlbmd0aCh0ZWNoX21lYW4pLAogICAgICAgICAgICAgaW5qX21lYW4gPSBtZWFuKHRlY2hfbWVhbiksCiAgICAgICAgICAgICBpbmpfc2QgPSBzZCh0ZWNoX21lYW4pLAogICAgICAgICAgICAgaW5qX3NlID0gaW5qX3NkL3NxcnQoaW5qX04pKQpzdGFuZGFyZHM0CgpgYGAKCiMjI1Bsb3QgYmVmb3JlIGFuZCBhZnRlciBwbG90cywgZmFjZXQgYnkgZXhwZXJpbWVudGFsIGRheQpgYGB7cn0Kc3RkX3Bsb3QgPC0gc3RhbmRhcmRzNCAlPiUgCiAgZ2dwbG90KGFlcyh4ID0gcGFydGljbGVfc2l6ZSwgeSA9IGlual9tZWFuLCBjb2xvcj1XaGVuKSkrCiAgZ2VvbV9saW5lKHNpemU9MikgKyB4bGltKDAsMzAwKSsgI2xpbmUgc2l6ZSwgeC1heGlzIHNjYWxlCiAgZ2VvbV9yaWJib24oYWVzKHltaW49aW5qX21lYW4taW5qX3NlLCB5bWF4PWlual9tZWFuK2lual9zZSksCiAgICAgICAgICAgICAgYWxwaGE9MC40LGZpbGwgPSBhbHBoYSgnZ3JleTEyJywgMC4yKSkgKyAjZXJyb3IgYmFycwogIHNjYWxlX3lfY29udGludW91cyhleHBhbmQ9YygwLDApKSsgI3NldCBib3R0b20gb2YgZ3JhcGgKICB4bGFiKCJQYXJ0aWNsZSBTaXplIikgKyAjIFggYXhpcyBsYWJlbAogIHlsYWIoIlxuTWVhbiBQYXJ0aWNsZSBDb25jZW50cmF0aW9uL21sXG4iKSArICMgWSBheGlzIGxhYmVsCiAgZ2d0aXRsZSgiTmFub3NpZ2h0IEhpc3RvZ3JhbSBvZlxuMTAwbm0gU3RhbmRhcmRzIikrICN0aXRsZQogIGxhYnMoY29sb3I9IkNvbmRpdGlvbiIpKyAjTGFiZWwgdGFibGUgdGl0bGUKICBmYWNldF93cmFwKH4gTmFub19kYXkpCgpzdGRfcGxvdAoKCiMgZ2dzYXZlKCJTdGFuZGFyZHNfaGlzdG9ncmFtX3Bsb3QucG5nIiwKIyAgICAgICAgaGVpZ2h0ID0gNSwgd2lkdGggPSA3LCBkcGkgPSAzMDAsIHVuaXRzPSAiaW4iKQpgYGAKCiMjIyBTdGFuZGFyZHMgcGFydGljbGUgY29uY2VudHJhdGlvbnMgZnJvbSBlYWNoIGV4cGVyaW1lbnRhbCBkYXkKYGBge3J9CnN0YW5kYXJkc19kZiA8LSBzdGFuZGFyZHM0ICU+JSAKICBncm91cF9ieShOYW5vX2RheSxXaGVuKSAlPiUgCiAgc3VtbWFyaXNlKHRvdGFsPXN1bShpbmpfbWVhbikpCgpzdGFuZGFyZHNfZGYKYGBgCgojIyNCYXIgZ3JhcGggb2Ygc3RhbmRhcmRzIHBhcnRpY2xlIGNvbmNlbnRyYXRpb25zCmBgYHtyfQpzdGFuZGFyZHNfYmFyIDwtIHN0YW5kYXJkc19kZiAlPiUgCiAgZ2dwbG90KGFlcyh4PU5hbm9fZGF5LHk9dG90YWwsZmlsbD1XaGVuKSkrCiAgZ2VvbV9jb2wocG9zaXRpb249ImRvZGdlIikrCiAgc2NhbGVfeV9jb250aW51b3VzKGV4cGFuZD1jKDAsMCkpKyAjc2V0IGJvdHRvbSBvZiBncmFwaAogIHhsYWIoIkV4cGVyaW1lbnRhbCBEYXkiKSArICMgWCBheGlzIGxhYmVsCiAgeWxhYigiXG5NZWFuIFBhcnRpY2xlIENvbmNlbnRyYXRpb24vbWxcbiIpICsgIyBZIGF4aXMgbGFiZWwKICBnZ3RpdGxlKCJOYW5vc2lnaHQgSGlzdG9ncmFtIG9mXG4xMDBubSBTdGFuZGFyZHMiKSsgI3RpdGxlCiAgbGFicyhjb2xvcj0iV2hlbiIpICNMYWJlbCB0YWJsZSB0aXRsZQoKc3RhbmRhcmRzX2JhcgoKIyBnZ3NhdmUoIlN0YW5kYXJkc19iYXJfcGxvdC5wbmciLAojICAgICAgICBoZWlnaHQgPSA1LCB3aWR0aCA9IDcsIGRwaSA9IDMwMCwgdW5pdHM9ICJpbiIpCmBgYAoKIyMjSW50cmFhc3NheSB2YXJpYWJpbGl0eQpgYGB7cn0KSW50cmEuYXNzYXlfY3YgPC0gc3RhbmRhcmRzX2RmICU+JSAKICBncm91cF9ieShOYW5vX2RheSkgJT4lIAogIHN1bW1hcmlzZShJbnRyYV9EYXlfTiA9IGxlbmd0aCh0b3RhbCksCiAgICAgICAgICAgIEludHJhX0RheV9tZWFuID0gbWVhbih0b3RhbCksCiAgICAgICAgICAgIEludHJhX0RheV9zZCA9IHNkKHRvdGFsKSwKICAgICAgICAgICAgSW50cmFfRGF5X3NlID0gSW50cmFfRGF5X3NkL3NxcnQoSW50cmFfRGF5X04pLAogICAgICAgICAgICBJbnRyYV9EYXlfY3YgPSBJbnRyYV9EYXlfc2QvSW50cmFfRGF5X21lYW4gKQpJbnRyYS5hc3NheV9jdgoKIyAjIFNhdmUgYXMgLmNzdgojIHdyaXRlX2NzdihJbnRyYS5hc3NheV9jdiwiSW50cmEuYXNzYXlfY3YuY3N2IikKYGBgCgojIyNJbnRlcmFzc2F5IHZhcmlhYmlsaXR5CmBgYHtyfQpJbnRlci5hc3NheV9jdiA8LSBJbnRyYS5hc3NheV9jdiAlPiUgCiAgc3VtbWFyaXNlKEludGVyX0RheV9OID0gbGVuZ3RoKEludHJhX0RheV9tZWFuKSwKICAgICAgICAgICAgSW50ZXJfRGF5X21lYW4gPSBtZWFuKEludHJhX0RheV9tZWFuKSwKICAgICAgICAgICAgSW50ZXJfRGF5X3NkID0gc2QoSW50cmFfRGF5X21lYW4pLAogICAgICAgICAgICBJbnRlcl9EYXlfc2UgPSBJbnRlcl9EYXlfc2Qvc3FydChJbnRlcl9EYXlfTiksCiAgICAgICAgICAgIEludGVyX0RheV9jdiA9IEludGVyX0RheV9zZC9JbnRlcl9EYXlfbWVhbiApCkludGVyLmFzc2F5X2N2CgojICMgU2F2ZSBhcyAuY3N2CiMgd3JpdGVfY3N2KEludGVyLmFzc2F5X2N2LCJJbnRlci5hc3NheV9jdi5jc3YiKQpgYGAKCgoKI1NhbXBsZSBhbmFseXNpcwoKIyMjQmFjayBjYWxjdWxhdGUgdGhlIG9yaWdpbmFsIGNvbmNlbnRyYXRpb24gb2YgdGhlIHNhbXBsZQpgYGB7cn0KZGF0YTIgPC0gZGF0YTIgJT4lIAogIG11dGF0ZShUcnVlX0NvdW50ID0gRGlsdXRpb25fZmFjdG9yKkNvdW50KQpkYXRhMgpgYGAKCgojIyMgQXZlcmFnZSB0aHJlZSB0ZWNobmljYWwgcmVhZGluZ3MKYGBge3J9CmRhdGEzIDwtIGRhdGEyICU+JSAKICBncm91cF9ieShwYXJ0aWNsZV9zaXplLFNhbXBsZV9JRCxEaWx1dGlvbl9mYWN0b3IsSW5qZWN0aW9uKSAlPiUgCiAgc3VtbWFyaXNlKCB0ZWNoX04gPSBsZW5ndGgoVHJ1ZV9Db3VudCksCiAgICAgICAgICAgICB0ZWNoX21lYW4gPSBtZWFuKFRydWVfQ291bnQpLAogICAgICAgICAgICAgdGVjaF9zZCA9IHNkKFRydWVfQ291bnQpLAogICAgICAgICAgICAgdGVjaF9zZSA9IHRlY2hfc2Qvc3FydCh0ZWNoX04pKQpkYXRhMwpgYGAKCiMjU3VtbWFyaXplIHNhbXBsZXMgYnkgaW5qZWN0aW9uIChhdmVyYWdlIGJvdGggaW5qZWN0aW9ucykKYGBge3IsIHdhcm5pbmc9RkFMU0V9CgpkYXRhNCA8LSBkYXRhMyAlPiUgCiAgZ3JvdXBfYnkocGFydGljbGVfc2l6ZSxTYW1wbGVfSUQsRGlsdXRpb25fZmFjdG9yKSAlPiUgCiAgc3VtbWFyaXNlKCBpbmpfTiA9IGxlbmd0aCh0ZWNoX21lYW4pLAogICAgICAgICAgICAgaW5qX21lYW4gPSBtZWFuKHRlY2hfbWVhbiksCiAgICAgICAgICAgICBpbmpfc2QgPSBzZCh0ZWNoX21lYW4pLAogICAgICAgICAgICAgaW5qX3NlID0gaW5qX3NkL3NxcnQoaW5qX04pKQpkYXRhNAoKIyBBdmVyYWdlIHRlY2huaWNhbCByZXBsaWNhdGVzIGFuZCBtZXJnZSB3aXRoIGtleQptZXJnZSA8LSBsZWZ0X2pvaW4oa2V5LGRhdGEzLCBieT0gIlNhbXBsZV9JRCIpCgptZXJnZQoKIyBTYXZlIGFzIC5jc3YKIyB3cml0ZV9jc3YobWVyZ2UsIlRlY2huaWNhbF9yZXBsaWNhdGVfYXZlcmFnZS5jc3YiKQogCiMgQXZlcmFnZSBpbmplY3Rpb24gcmVwbGljYXRlcyBhbmQgbWVyZ2Ugd2l0aCBrZXkKbWVyZ2UxIDwtIGxlZnRfam9pbihrZXksZGF0YTQsIGJ5PSAiU2FtcGxlX0lEIikKCm1lcmdlMQoKIyAjU2F2ZSBhcyAuY3N2CiMgd3JpdGVfY3N2KG1lcmdlMSwiSW5qZWN0aW9uX3JlcGxpY2F0ZV9hdmVyYWdlLmNzdiIpCgpgYGAKCgoKI1F1aWNrIHZpc3VhbGl6YXRpb25zCgojIyMgR3JhcGhpbmcgYWxsIHNhbXBsZXMKYGBge3IsIHdhcm5pbmc9RkFMU0V9CnNhbXBsZV9wbG90IDwtIG1lcmdlICU+JQogIGdncGxvdChhZXMoeD1wYXJ0aWNsZV9zaXplLCB5PXRlY2hfbWVhbixjb2xvcj1JbmplY3Rpb24gKSkrICNwbG90CiAgZ2VvbV9yaWJib24oYWVzKHltaW49dGVjaF9tZWFuLXRlY2hfc2UsCiAgICAgICAgICAgICAgICAgIHltYXg9dGVjaF9tZWFuK3RlY2hfc2UpLAogICAgICAgICAgICAgICAgICBhbHBoYT0wLjIsZmlsbCA9IGFscGhhKCdncmV5MTInLCAwLjIpKSArICNlcnJvciBiYXJzCiAgZ2VvbV9saW5lKHNpemU9Mi4wLCBhbHBoYSA9IDAuOCkgKyB4bGltKDAsNTAwKSsgI2xpbmUgc2l6ZSwgeC1heGlzIHNjYWxlCiAgc2NhbGVfeV9jb250aW51b3VzKGV4cGFuZD1jKDAsMCkpKyAjc2V0IGJvdHRvbSBvZiBncmFwaAogIHhsYWIoIlBhcnRpY2xlIFNpemUiKSArICMgWCBheGlzIGxhYmVsCiAgeWxhYigiXG5NZWFuIFBhcnRpY2xlIENvbmNlbnRyYXRpb24vbWxcbiIpICsgIyBZIGF4aXMgbGFiZWwKICBnZ3RpdGxlKCJOYW5vc2lnaHQgSGlzdG9ncmFtIG9mXG5oQ1RCUyB0cmVhdGVkIHdpdGggQlBTIikrICN0aXRsZQogIGxhYnMoY29sb3I9IkluamVjdGlvbiIpKyAjTGFiZWwgdGFibGUgdGl0bGUKICBmYWNldF9ncmlkKFRpbWUgfiBUcmVhdG1lbnQpCiAgIyBnZW9tX3ZsaW5lKHhpbnRlcmNlcHQgPSAyMDApKwogICMgYW5ub3RhdGUoInRleHQiLCB4PSAzNTAsIHkgPSAxRTgsIGxhYmVsPSAiMjAwbm0iKQoKc2FtcGxlX3Bsb3QKCiMgZ2dzYXZlKCJOYW5vc2lnaHRfU2FtcGxlX0hpc3RvZ3JhbS5wbmciLCBwbG90ID0gc2FtcGxlX3Bsb3QsCiMgICAgICAgIGhlaWdodCA9IDEwLCB3aWR0aCA9IDE0LCBkcGkgPSAyMDAsIHVuaXRzPSAiaW4iKQpgYGAKIyMjSW50ZXJhY3RpdmUgUGxvdApgYGB7cn0KZ2dwbG90bHkoc2FtcGxlX3Bsb3QpCmBgYAoKCgojIyMgUGFydGljbGUgY29uY2VudHJhdGlvbiB2YWx1ZXMgZm9yIGVhY2ggb2YgdGhlIHNhbXBsZXMKYGBge3J9Cm1lcmdlMiA8LSBtZXJnZTEgJT4lIAogIGdyb3VwX2J5KFRpbWUsIFRyZWF0bWVudCwgVm9sdW1lKSAlPiUgCiAgc3VtbWFyaXNlKHBhcnRpY2xlX2NvbmM9c3VtKGlual9tZWFuKSkKbWVyZ2UyCmBgYAoKIyMjIENvcnJlY3QgZm9yIHJlc3VzcGVuc2lvbiB2b2x1bWUKYGBge3J9Cm1lcmdlMyA8LSBtZXJnZTIgJT4lIAogIG11dGF0ZShwYXJ0aWNsZV9jb3VudCA9IChWb2x1bWUvMTAwMCkqcGFydGljbGVfY29uYywgIyBDcmVhdGUgbmV3IGNvbHVtbiB3aXRoIG51bWJlciBvZiBwYXJ0aWNsZXMKICAgICAgICAgY29ycmVjdGVkX3BhcnRpY2xlX2NvbmMgPSAocGFydGljbGVfY29uYy8xRTkpKSAjIENyZWF0ZSBuZXcgY29sdW1uIHdpdGggY29ycmVjdCBwYXJ0aWNsZSBjb25jZW50cmF0aW9uCm1lcmdlMwoKIyBTYXZlIGFzIC5jc3YKIyB3cml0ZV9jc3YobWVyZ2UzLCJBZGp1c3RlZF9wYXJ0aWNsZV9jb25jZW50cmF0aW9uLmNzdiIpCgpgYGAKCgojIyNCYXJwbG90CmBgYHtyLCB3YXJuaW5nPVRSVUV9CnBsb3QxIDwtIG1lcmdlMyAlPiUgCiAgZ2dwbG90KGFlcyh4ID0gVHJlYXRtZW50LCB5ID0gY29ycmVjdGVkX3BhcnRpY2xlX2NvbmMsIGZpbGwgPSBUcmVhdG1lbnQpKSArCiAgZ2VvbV9iYXIoYWVzKHRleHQgPSBwYXN0ZSgiUGFydGljbGUgQ29uY2VudHJhdGlvbjoiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgY29ycmVjdGVkX3BhcnRpY2xlX2NvbmMpKSwKICAgICAgICAgICBzdGF0PSJpZGVudGl0eSIsIHBvc2l0aW9uID0gImRvZGdlIikrCiAgeGxhYigiXG5UcmVhdG1lbnRcbiIpICsgIyBYIGF4aXMgbGFiZWwKICB5bGFiKCJcbk1lYW4gVmVzc2ljbGUgQ29uY2VudHJhdGlvblxuKDEwXjkgcGFydGljbGVzLyBtbClcbiIpICsgIyBZIGF4aXMgbGFiZWwKICBnZ3RpdGxlKCJFZmZlY3Qgb2YgQlBTIG9uIEV4dHJhY2VsbHVsYXIgVmVzc2ljbGVcblJlbGVhc2Ugb2YgaENUQnNcbiIpKwogIHNjYWxlX3lfY29udGludW91cyhicmVha3MgPSBzZXEoMCwxNCwyKSwKICAgICAgICAgICAgICAgICAgICAgbGltaXRzID0gYygwLDE0KSwKICAgICAgICAgICAgICAgICAgICAgZXhwYW5kID0gYygwLDApKSsgIyBzZXQgYm90dG9tIG9mIGdyYXBoCiAgbGFicyhjb2xvcj0iQ29uZGl0aW9uIikrICMgTGFiZWwgdGFibGUgdGl0bGUKICBmYWNldF93cmFwKH5UaW1lKQoKICAKcGxvdDEKCiMgZ2dzYXZlKCJCUFNfdHJlYXRlZF9oQ1RCc180OF85Nl9mYWNldF9wbG90LnBuZyIsCiMgICAgICAgaGVpZ2h0ID0gOCwgd2lkdGggPSAxMCwgZHBpID0gNjAwLCB1bml0cz0gImluIikKYGBgCiMjSW50ZXJhY3RpdmUgUGxvdApgYGB7cn0KZ2dwbG90bHkocGxvdDEpCmBgYAoKIyMjQmFycGxvdApgYGB7cn0KcGxvdDIgPC0gbWVyZ2UzICU+JQogIGdyb3VwX2J5KFRpbWUpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSBUcmVhdG1lbnQsIHkgPSBjb3JyZWN0ZWRfcGFydGljbGVfY29uYywgZmlsbCA9IFRpbWUgKSkgKwogIGdlb21fYmFyKHBvc2l0aW9uID0gImRvZGdlIiwgc3RhdCA9ICJpZGVudGl0eSIpKwogIHhsYWIoIlxuVHJlYXRtZW50XG4iKSArICMgWCBheGlzIGxhYmVsCiAgeWxhYigiXG5NZWFuIFZlc3NpY2xlIENvbmNlbnRyYXRpb25cbigxMF45IHBhcnRpY2xlcy8gbWwpXG4iKSArICMgWSBheGlzIGxhYmVsCiAgZ2d0aXRsZSgiRWZmZWN0IG9mIEJQUyBvbiBFeHRyYWNlbGx1bGFyIFZlc3NpY2xlXG5SZWxlYXNlIG9mIGhDVEJzXG4iKSsKICBzY2FsZV95X2NvbnRpbnVvdXMoYnJlYWtzID0gc2VxKDAsMTQsMiksCiAgICAgICAgICAgICAgICAgICAgIGxpbWl0cyA9IGMoMCwxNCksCiAgICAgICAgICAgICAgICAgICAgIGV4cGFuZCA9IGMoMCwwKSkrICMgc2V0IGJvdHRvbSBvZiBncmFwaAogIGxhYnMoZmlsbD0gIlRpbWUgKGhyKSIpCgpwbG90MgoKICMgZ2dzYXZlKCJCUFNfdHJlYXRlZF9oQ1RCc180OF85Nl9wbG90LnBuZyIsCiAjICAgICAgIGhlaWdodCA9IDUsIHdpZHRoID0gNywgZHBpID0gNjAwLCB1bml0cz0gImluIikKYGBgCiMjIEludGVyYWN0aXZlIFBsb3QKYGBge3J9CmdncGxvdGx5KHBsb3QyKQpgYGAKCgoKIyBTdGF0aXN0aWNzCgpgYGB7cn0KZml0IDwtIGFvdihjb3JyZWN0ZWRfcGFydGljbGVfY29uYyB+IFRpbWUgKiBUcmVhdG1lbnQgLGRhdGE9bWVyZ2UzKQoKdGlkeShmaXQpCmBgYAoK